```
* James Small & William Elder
 * OS.h
 * 6/7/13
 ^{\star} This program is the main OS class that uses an assembler and virtual machine
 * objects to run our programs. This class takes in the program files, converts
 * them to object code, loads them all into memory, and then proceeds to run them
 * all using context switches to allow all of the programs to run concurrently,
 * allowing each program to have access to the cpu at some point.
 #ifndef OS H
#define OS H
#include "Assembler.h"
#include "VirtualMachine.h"
#include "PCB.h"
#include <list>
#include <queue>
#include <string>
#include <cstring>
using namespace std;
class OS {
private:
    //variables
   Assembler as;
   VirtualMachine vm;
   list<PCB *> jobs;
    queue<PCB *> readyQ, waitQ;
   PCB * running;
   int contextSwitchClock, idleClock;
   int MULTIPROGCAP;
   list<int> unusedFrames;
   fstream readProgs;
   bool lessThanMultiProgCap;
   int pageReplaced;
   // internal functions
   void getPCBFromVM();
   void givePCBToVM();
   void outputErrorCode(string s);
   bool done();
   void readIn(string line);
   void replaceAlgorithm();
public:
   void start(int lruValue);
   OS();
   ~OS();
};
#endif
```

```
* James Small & William Elder
* OS.h
* 6/7/13
^{\star} This program is the main OS class that uses an assembler and virtual machine
* objects to run our programs. This class takes in the program files, converts
^{\star} them to object code, loads them all into memory, and then proceeds to run them
* all using context switches to allow all of the programs to run concurrently,
* allowing each program to have access to the cpu at some point.
#include "OS.h"
OS::OS()
   contextSwitchClock = idleClock = pageReplaced = 0;
   MULTIPROGCAP = 5;
   for (int i = 0; i < 32; i++)
      unusedFrames.push back(i);
   lessThanMultiProgCap = false;
}
OS::~OS()
   readProgs.close();
   system("rm progs");
void OS::start(int lruValue)
update to take a string that determines if lru or fifo. Based on that update, master variable
in VM
that controls if lru or fifo
*****/
******/
   if (lruValue == 1)
      vm.lru = true;
   else
      vm.lru = false;
// read in all s files
   system("ls *.s > progs");
   readProgs.open("progs", ios::in);
   if (!readProgs.is_open()) {
      cout << "Progs failed to open.\n";</pre>
      exit(1);
   }
   string line;
   for (int i = 0; i < MULTIPROGCAP && !readProgs.eof(); i++)</pre>
      getline(readProgs, line);
      if (readProgs.eof()) {
         lessThanMultiProgCap = true;
         break;
      }
```

```
readIn(line);
} // end reading into memory
// copy first item in readyQ to running , load pcb data to vm
running = readyQ.front();
readyQ.pop();
givePCBToVM();
int code = 0;
while (!done()) // main process loop
    if (lessThanMultiProgCap) {
        getline(readProgs, line);
        if (!readProgs.eof()) {
            lessThanMultiProgCap = false;
            readIn(line);
    }
    code = 0;
    vm.run(); // runs current proceess in vm, returns error code stored for later
    // get return code from status register
    code = vm.sr;
    code = code & 0xe0;
    code >>= 5;
    // update clocks for context switch
    vm.masterClock += 5;
    contextSwitchClock += 5;
    // update all processes that aren't terminated with addition to context switch time
    for (list<PCB *>::iterator it = jobs.begin(); it != jobs.end(); it++)
        if(!strcmp((*it)->state.c str(), "terminated"))
            (*it) -> contextSwitchTime += 5;
    // First item in main loop
    // move items from waitQ into readyQ if interrupt time hit
    if(!waitQ.empty()) {
        while (waitQ.front()->interruptTime <= vm.masterClock) {</pre>
            PCB * temp = waitQ.front();
            waitQ.pop();
            readyQ.push(temp);
            temp->state = "ready";
            if(waitQ.empty())
                break;
    // Second item in main loop
    // Update waiting time for all items in ready queue with latest cycle time from vm.run()
    for (list<PCB *>::iterator it = jobs.begin(); it != jobs.end(); it++)
        if(strcmp((*it)->state.c_str(), "ready"))
            (*it) -> waitingTime += vm.clock;
    if(!(running == NULL)) {
        // update master clock and cpu time
        vm.masterClock += vm.clock;
        running->cpuTime += vm.clock;
```

```
// main switch to handle return code, if terminated set, output errorcode.
            switch (code) {
                case 0:
                    int bit10 = vm.sr \& 0x400;
                    bit10 = bit10 >> 10;
                    int frameNumber;
                    if (bit10 == 0) // time slice
                        getPCBFromVM();
                        readyQ.push(running);
                    else // page fault
                        if (unusedFrames.empty())
                            replaceAlgorithm();
                        frameNumber = unusedFrames.front();
                        unusedFrames.pop front();
                        int baseNew = 0;
                        int limitNew = 0;
                        if (vm.pageNumberToGet == -1) {
                            exit(1);
                        int temp = vm.loadIntoMemory(running->originalfilename,
vm.pageNumberToGet, frameNumber, &baseNew, &limitNew);
                        if (temp == 1) {
                            running->state = "terminated";
                            outputErrorCode("Out-of-Bound Reference");
                            lessThanMultiProgCap = true;
                        } else {
                            vm.frames[vm.pageNumberToGet] = frameNumber;
                            vm.valid[vm.pageNumberToGet] = 1;
                            // pc = next logical address
                            vm.base = vm.frames[vm.pageNumber(vm.pc)] * vm.FRAME SIZE;
                            vm.pc = vm.frames[vm.pageNumber(vm.pc)] * vm.FRAME SIZE +
vm.offsetNumber(vm.pc);
                            waitQ.push(running);
                            running->state = "waiting";
                            getPCBFromVM();
                            running->interruptTime = vm.masterClock + 35;
                    }
                    break;
                case 1:
                    running->state = "terminated";
                    lessThanMultiProgCap = true;
                    break;
                case 2:
                    running->state = "terminated";
                    outputErrorCode("Out-of-Bound Reference");
```

```
lessThanMultiProgCap = true;
    break;
case 3:
   running->state = "terminated";
    outputErrorCode("Stack Overflow");
    lessThanMultiProgCap = true;
    break;
case 4:
    running->state = "terminated";
    outputErrorCode("Stack Underflow");
    lessThanMultiProgCap = true;
    break;
case 5:
   running->state = "terminated";
    outputErrorCode("Invalid OPCode");
    lessThanMultiProgCap = true;
    break;
case 6:
    int readIn;
    running->readIntoRegister >> readIn;
    int readReg = vm.sr;
    readReg &= 0x300;
    readReg >>= 8;
    if(readReg < 0 || readReg > 3) {
        outputErrorCode("Invalid IO Register");
        break:
    }
    vm.reg[readReg] = readIn;
    vm.masterClock++;
    running->ioTime += 27;
    running->cpuTime++;
    waitQ.push(running);
    running->state = "waiting";
    getPCBFromVM();
    running->interruptTime = vm.masterClock + 28;
    break;
}
case 7:
    int writeToReg = vm.sr;
    writeToReg &= 0x300;
    writeToReg >>= 8;
    int temp = vm.reg[writeToReg] & 0x8000;
    if (temp)
        temp = vm.reg[writeToReg] | 0xffff0000;
    else
        temp = vm.reg[writeToReg] & 0xffff;
    running->writeToRegister << temp << endl;
    vm.masterClock++;
    running->ioTime += 27;
    running->cpuTime++;
    waitQ.push(running);
    running->state = "waiting";
    getPCBFromVM();
```

```
running->interruptTime = vm.masterClock + 28;
                    break;
                default:
                    outputErrorCode("Invalid Return Code");
                    running->state = "terminated";
                    lessThanMultiProgCap = true;
                    break;
            running = NULL;
        } // end if running loop
        // Third item in main loop
        // if all items in waitQ, update vm.masterClock to get first item out of waitQ
        if(readyQ.empty() && !waitQ.empty()) {
            int temp = waitQ.front()->interruptTime - vm.masterClock;
            vm.masterClock += temp;
            idleClock += temp;
            readyQ.push(waitQ.front());
            waitQ.front()->state = "ready";
            waitQ.pop();
            // update idle time for all non terminated process in jobs list
            for (list<PCB *>::iterator it = jobs.begin(); it != jobs.end(); it++)
                if(!strcmp((*it)->state.c_str(), "terminated"))
                    (*it)->idleTime += temp;
        }
        // move next process from readyQ into running state, copy over pcb contents
        if(!readyQ.empty()) {
            running = readyQ.front();
            running->state = "running";
            readyQ.pop();
            givePCBToVM();
        }
    } // end main process loop
    // run program accounting data
    // calculate turnaround time for all processes
    for (list<PCB *>::iterator it = jobs.begin(); it != jobs.end(); it++)
        (*it)->turnaroundTime = (*it)->cpuTime + (*it)->waitingTime + (*it)->ioTime + (*it)->
idleTime + (*it)->contextSwitchTime;
    // run system accounting data
    int systemTime = contextSwitchClock + idleClock; // calculate system time
    double floatingPointMasterClock = vm.masterClock; // calculate CPU Utilization
    double cpuUtilization = (floatingPointMasterClock - idleClock) / floatingPointMasterClock;
// calculate CPU Utilization
    int allCpuTime = 0;
    // iterate through all processes to add cpuTime
    for (list<PCB *>::iterator it = jobs.begin(); it != jobs.end(); it++)
        allCpuTime += (*it)->cpuTime;
    double userCpuUtilization = allCpuTime / floatingPointMasterClock; // calculate user cpu
ulitizaton.
    double throughPut = jobs.size() / (floatingPointMasterClock / 10000.0); // calculate
```

```
throughput
```

```
calculate hit ratio and add to output for each file below
************************************
*****/
    // iterate through pcb list and output throughput and vm utilization to all files
    for (list<PCB *>::iterator it = jobs.begin(); it != jobs.end(); it++) {
        fstream writeAccounting;
        writeAccounting.open((*it)->writefilename.c str(), ios::in | ios::out | ios::ate);
        if (!writeAccounting.is open()) {
            cout << (*it)->writefilename << " failed to open.\n";</pre>
            exit(1);
        writeAccounting << "\nProcess Specific Accounting Data\n\n";
        writeAccounting << "CPU Time = " << (*it)->cpuTime << " cycles\n";</pre>
        writeAccounting << "Waiting Time = " << (*it)->waitingTime << " cycles\n";</pre>
        writeAccounting << "Turnaround Time = " << (*it)->turnaroundTime << " cycles\n";</pre>
        writeAccounting << "IO Time = " << (*it)->ioTime << " cycles\n";</pre>
        writeAccounting << "Largest Stack Size = " << (*it)->largestStackSize << endl;</pre>
        writeAccounting << "\nSystem Specific Accounting Data\n\n";</pre>
        writeAccounting << "System Time = " << systemTime << " cycles\n";</pre>
        writeAccounting << "System CPU Utilization = " << cpuUtilization * 100 << "%\n";
        writeAccounting << "User CPU Utilization = " << userCpuUtilization * 100 << "%\n";</pre>
        \label{eq:writeAccounting} << \text{"ThroughPut} = \text{"} << \text{throughPut} << \text{"processes per second} \\ \text{"};
        writeAccounting.close();
    }
    // make sure all .st files are deleted.
    system("rm *.st");
void OS::givePCBToVM()
    //copy contents of PCB running into VM, including stack information
    vm.pc = running->pc;
    vm.sr = running->sr;
    vm.sp = running->sp;
    vm.base = running->base;
    vm.limit = running->limit;
    vm.inputFileName = running->readfilename;
    vm.originalFileName = running->originalfilename;
    for (int i = 0;i <= vm.REG FILE SIZE - 1;i++)
        vm.reg[i] = running->reg[i];
    if (running->sp < 256) {
        int spTemp = running->sp;
        while (spTemp < 256)
```

```
int frameNumberToOverwrite = vm.pageNumber(spTemp);
            vm.copyFromMemory(frameNumberToOverwrite,&jobs);
            spTemp += 6;
        fstream readProgs;
        readProgs.open(running->stfilename.c_str(), ios::in);
        if (!readProgs.is open()) {
            cout << running->stfilename << " failed to open.\n";</pre>
            exit(1);
        }
        int stackPointer = running->sp;
        string line;
        getline(readProgs, line);
        while (!readProgs.eof()) {
            vm.mem[stackPointer++] = atoi(line.c str());
            getline(readProgs, line);
        }
    }
    // copy to TLB
    vm.frames = running->frames;
    vm.valid = running->valid;
    vm.modified = running->modified;
        vm.accessRegisters[vm.frames[vm.pageNumber(vm.physicalToLogical(vm.pc))]] =
vm.masterClock;
void OS::getPCBFromVM()
    //copy contents of VM into running PCB, including stack information
    running->pc= vm.pc;
    running->sr= vm.sr;
    running->sp=vm.sp;
    running->base= vm.base;
   running->limit= vm.limit;
   vm.osJobs = &jobs;
    for (int i = 0; i \le vm.REG FILE SIZE - 1; i++) {
        running->reg[i] = vm.reg[i];
    }
   if (running->sp < 256) {
        int spTemp = running->sp;
       while (spTemp < 256)
            vm.inverted[vm.pageNumber(spTemp)] = "NULL";
```

```
spTemp += 6;
    }
    if(running->sp < 256)
        fstream readProgs;
        readProgs.open(running->stfilename.c str(), ios::out);
        if (!readProgs.is open()) {
             cout << running->stfilename << " failed to open.\n";</pre>
            exit(1);
        int stackPointer = running->sp;
        int temp = 256 - stackPointer;
        if(running->largestStackSize < temp)</pre>
            running->largestStackSize = temp;
        while (stackPointer<256) {</pre>
             readProgs<<vm.mem[stackPointer++]<<endl;</pre>
   // TLB to PCB
    running->frames = vm.frames;
    running->valid = vm.valid;
    running->modified = vm.modified;
void OS::outputErrorCode(string s)
    fstream output;
    output.open(running->writefilename.c_str(), ios::in | ios::out | ios::ate);
    output << "\nError Code = " << s << endl << endl;</pre>
    output.close();
// check if all processes have been terminated
bool OS::done()
    if(readyQ.empty() && waitQ.empty() && !running)
        return true;
    return false;
void OS::readIn(string line)
        as.assemble(line); // assemble each file
        //create PCB for each file, put in jobs list, and put in ready queue
        PCB * p = new PCB;
        jobs.push back(p);
        readyQ.push(p);
        p->state = "ready";
        string fileName = line.erase(line.length() -2, 2);
        // set file names for input and output
        p->originalfilename = fileName + ".o";
```

```
p->readfilename = fileName + ".in";
       p->writefilename = fileName + ".out";
       p->stfilename = fileName + ".st";
       // Open input and output file streams in PCB
       p->openReadInFileStream();
       p->openWriteOutFileStream();
       //read into memory, modify PCB with values for base and limit
       int base, limit;
       //calculate page and frame number to use
       if (unusedFrames.empty())
               replaceAlgorithm();
       int frameNumber = unusedFrames.front();
       unusedFrames.pop_front();
       int pageNum = vm.pageNumber(p->pc);
       int temp = vm.loadIntoMemory(p->originalfilename, pageNum, frameNumber, &base, &limit);
       p->base = base;
       p->limit = limit;
       //set page table
       p->frames[pageNum] = frameNumber;
       p->valid[pageNum] = 1;
       p->pc = frameNumber * vm.FRAME_SIZE + vm.offsetNumber(p->pc);
void OS::replaceAlgorithm()
   int oldest = 0;
   if (!strcmp(vm.inverted[oldest].c_str(),"Stack"))
       oldest = 1;
   for (int i = oldest + 1; i < vm.accessRegisters.size();i++)
       if (vm.accessRegisters[i] < vm.accessRegisters[oldest])</pre>
           if (strcmp(vm.inverted[i].c_str(),"Stack"))
               oldest = i;
       }
   }
   vm.copyFromMemory(oldest,&jobs);
   unusedFrames.push back(oldest);
   pageReplaced++;
// main to run OS
int main(int argc, char * argv[])
update to allow the passing in of a variable to determine lru or fifo and pass to .start()
```

}

```
* James Small & William Elder
  VirtualMachine.H
 * 4/17/13
 * This program simulates a 16 bit virtual machine. It reads in machine code
 * created by the Assembler program, pulls them all into memory, and then executes
 * each instruction one at a time.
 ^{\star} Modified 5/7/13: Add the ability to run multiple programs at a time. Main run
 * function was modified to allow this. Also added the ability to load programs
 * into memory before beginning to run the programs.
 *************************************
#ifndef VIRTUALMACHINE H
#define VIRTUALMACHINE H
#include <iostream>
#include <fstream>
#include <string>
#include <map>
#include <vector>
#include <cstdlib>
#include <stdexcept>
#include <list>
#include "PCB.h"
#include <cstring>
using namespace std;
class format1 {
public:
   unsigned UNUSED:6;
   unsigned RS:2;
   unsigned I:1;
   unsigned RD:2;
   unsigned OP:5;
class format2 {
public:
   unsigned ADDR:8;
   unsigned I:1;
   unsigned RD:2;
   unsigned OP:5;
};
class format3 {
public:
   int CONST:8;
   unsigned I:1;
   unsigned RD:2;
   unsigned OP:5;
};
union instruction {
   int i;
   format1 f1;
   format2 f2;
   format3 f3;
};
class VirtualMachine {
private:
   //typedefs
   typedef void(VirtualMachine::*FP)();
   // Variables
   static const int REG FILE SIZE = 4;
   static const int MEM_SIZE = 256;
   static const int FRAME SIZE = 8;
   vector < int > reg;
   vector < int > mem;
```

```
instruction finalcode;
    int pc, ir, sr, sp, base, limit, clock, nextMemoryLocation;
    map<int, FP> instr;
    string inputFileName, readFileName, writeFileName, originalFileName;
    bool execute;
    vector <int> frames;
    vector <int> valid;
    vector <int> modified;
    vector <int> accessRegisters;
    bool lru;
    vector <string> inverted;
    int pageNumberToGet;
    int masterClock;
    list<PCB *> * osJobs;
// Operations
    void setCarryBit();
    void setRightCarryBit();
    void setOverflow();
    int getCarryBit();
    void setGreaterThan();
    int getGreaterThan();
    void resetGreaterThan();
    void setLessThan();
    int getLessThan();
    void resetLessThan();
    void setEqualTo();
    int getEqualTo();
    void resetEqualTo();
    void returnCode(int i);
    void readOrWrite(int i);
    int pageNumber(int addr);
    int offsetNumber(int addr);
    bool isItInMemory(int ADDR);
    int physicalToLogical(int ADDR);
// Instruction Functions
    void load();
    void loadi();
    void store();
    void add();
   void addi();
   void addc();
   void addci();
   void sub();
   void subi();
   void subc();
   void subci();
   void and();
   void andi();
   void xor();
   void xori();
   void compl();
   void shl();
   void shla();
   void shr();
   void shra();
   void compr();
   void compri();
   void getstat();
   void putstat();
   void jump();
   void jumpl();
   void jumpe();
   void jumpg();
   void call();
   void return();
   void read();
   void write();
```

```
void halt();
void noop();

public:
    VirtualMachine();
    void run();
    int loadIntoMemory(string originalFileName, int pageNumber, int frameNumber, int * progBase,
int * progLimit);
    void copyFromMemory(int frame, list<PCB *> *jobs);
    friend class OS;
};

#endif
```

```
* James Small & William Elder
 * VirtualMachine.cpp
 * 4/17/13
 * This program simulates a 16 bit virtual machine. It reads in machine code
 * created by the Assembler program, pulls them all into memory, and then executes
 * each instruction one at a time.
 * Modified 5/7/13: Add the ability to run multiple programs at a time. Main run
 * function was modified to allow this. Also added the ability to load programs
 * into memory before beginning to run the programs.
 *******************************
#include "VirtualMachine.h"
VirtualMachine::VirtualMachine()
   reg = vector <int> (REG FILE SIZE);
   mem = vector <int> (MEM SIZE);
   execute = true;
   nextMemoryLocation = 0;
   pageNumberToGet = -1;
   masterClock = 0;
   accessRegisters = vector <int> (32); // for 32 frames
   for (int i = 0;i < accessRegisters.size();i++)</pre>
       accessRegisters[i] = -1;
       inverted = vector <string> (32); // for 32 frames
   for (int i = 0;i < inverted.size();i++)</pre>
       inverted[i] = "NULL";
   instr[0] = &VirtualMachine::load;
   instr[1] = &VirtualMachine::store;
   instr[2] = &VirtualMachine::add;
   instr[3] = &VirtualMachine::addc;
   instr[4] = &VirtualMachine::sub;
   instr[5] = &VirtualMachine::subc;
   instr[6] = &VirtualMachine:: and;
   instr[7] = &VirtualMachine::_xor;
   instr[8] = &VirtualMachine:: compl;
   instr[9] = &VirtualMachine::shl;
   instr[10] = &VirtualMachine::shla;
   instr[11] = &VirtualMachine::shr;
   instr[12] = &VirtualMachine::shra;
   instr[13] = &VirtualMachine::compr;
   instr[14] = &VirtualMachine::getstat;
   instr[15] = &VirtualMachine::putstat;
   instr[16] = &VirtualMachine::jump;
   instr[17] = &VirtualMachine::jumpl;
   instr[18] = &VirtualMachine::jumpe;
   instr[19] = &VirtualMachine::jumpg;
   instr[20] = &VirtualMachine::call;
   instr[21] = &VirtualMachine:: return;
   instr[22] = &VirtualMachine::read;
   instr[23] = &VirtualMachine::write;
   instr[24] = &VirtualMachine::halt;
   instr[25] = &VirtualMachine::noop;
int VirtualMachine::loadIntoMemory(string originalFileName, int pageNumber, int frameNumber, int
* progBase, int * progLimit)
   // read into memory, increase limit
   fstream readIntoMemory;
   readIntoMemory.open(originalFileName.c str(), ios::in);
```

```
if (!readIntoMemory.is_open()) {
        cout << inputFileName << " failed to open.\n";</pre>
   nextMemoryLocation = frameNumber * FRAME SIZE;
   string line;
    for (int i = 0; i < pageNumber * FRAME SIZE;i++) {</pre>
        getline(readIntoMemory, line);
 /*
    if (readIntoMemory.eof()) {
        returnCode(2);
       readIntoMemory.close();
        return 1;
    *progBase = nextMemoryLocation;
    for (int i = 0; i < FRAME SIZE; i++)
        if (readIntoMemory.eof()) {
            while (i < FRAME SIZE) {
                mem[nextMemoryLocation++] = 0;
                i++;
            break;
        getline(readIntoMemory, line);
        mem[nextMemoryLocation++] = atoi(line.c_str());
        int temp = atoi(line.c str());
    *progLimit = 8;
    readIntoMemory.close();
    inverted[frameNumber] = originalFileName;
    if (!lru)
        accessRegisters[frameNumber] = masterClock;
     return 0;
}
void VirtualMachine::copyFromMemory(int frame, list<PCB*>* jobs)
    bool needToWriteToDisk = false;
    string updateFileName;
    string process = inverted[frame];
    int page = 0;
    inverted[frame] = "NULL";
    if (!strcmp(process.c_str(),originalFileName.c str()))
        for (int i = 0; i < frames.size(); i++)
            if (frames[i] == frame) {
```

```
page = i;
        break;
    }
    updateFileName = originalFileName;
    if(modified[page] == 1) {
        needToWriteToDisk = true;
        modified[page] = 0;
    }
    valid[page] = 0;
else {
    for (list<PCB *>::iterator it = jobs->begin(); it != jobs->end(); it++)
        if(!strcmp((*it)->originalfilename.c str(), process.c str()))
            for (int i = 0; i < (*it) \rightarrow frames.size(); i++)
                 if ((*it) \rightarrow frames[i] == frame) {
                 page = i;
                 break;
            updateFileName = (*it)->originalfilename;
            if((*it)->modified[page] == 1) {
                 needToWriteToDisk = true;
                 (*it)->modified[page] = 0;
             (*it) - valid[page] = 0;
        }
    }
}
if (needToWriteToDisk)
// copy back to disk
    fstream readProgs;
    readProgs.open(updateFileName.c str(), ios::in);
    string newFileName = updateFileName + "2";
    fstream newFile;
    newFile.open(newFileName.c_str(), ios::out);
    if (!readProgs.is_open()) {
        cout << "Progs failed to open.\n";
    }
    if (!newFile.is open()) {
        cout << "Progs failed to open.\n";</pre>
    }
    string line;
    getline(readProgs, line);
```

```
int i = 0;
        int count = FRAME SIZE * page;
        int frameCount = FRAME SIZE * frame;
        while (!readProgs.eof() && i < count)</pre>
            newFile << line << endl;
            getline(readProgs, line);
            i++;
        }
        for(int j = 0;j < FRAME SIZE && mem[frameCount] != 0;j++)</pre>
            newFile << mem[frameCount] << endl;</pre>
            getline(readProgs, line);
            frameCount++;
        while (!readProgs.eof())
            newFile << line << endl;</pre>
            getline(readProgs, line);
        readProgs.close();
        newFile.close();
        string removeFile = "rm " + updateFileName;
        system(removeFile.c str());
        int result= rename(newFileName.c_str() , updateFileName.c_str() );
}
void VirtualMachine::run() {
   clock = 0;
   execute = true;
   sr &= 0xfffffffff; // resets return code to 0
   sr = sr & 0xfffffbff; // resets bit 10 to 0
   pageNumberToGet = -1;
   int logicalPc = physicalToLogical(pc);
   if (!valid[pageNumber(logicalPc)]) {
       pageNumberToGet = pageNumber(logicalPc);
       pc = logicalPc;
       sr = sr \mid 0x400;
       execute = false;
       return;
   // execute loop
   while (execute)
       if (pc < base)
            exit(1);
       if (pc >= base + limit) // page fault
            int pageNum = physicalToLogical(pc - 1);
```

```
pageNum = pageNumber(pageNum);
            pageNum++;
            if (valid[pageNum] == 0) { // page not in memory
                pageNumberToGet = pageNum;
                pc = pageNum * FRAME SIZE;
                 sr = sr \mid 0x400;
                 execute = false;
                 continue;
            } else { // page in memory
                pc = frames[pageNum] * FRAME_SIZE;
                base = pc;
                if(lru)
                     accessRegisters[frames[pageNumber(physicalToLogical(pc))]] = masterClock;
            }
    }
        ir = mem[pc++];
        finalcode.i = ir;
        (this->*instr[finalcode.f1.OP])();
        if(clock >= 15)
            execute = false;
    } // end while loop
} // run function
void VirtualMachine::setCarryBit()
    int temp = reg[finalcode.f1.RD] & 0x10000; // get 17th bit only
    temp = temp >> 17; // right shift 17
    if (sr & 0x1) // check if sr carry bit is 1, if so, subtract 1 from it to reset to 0
        sr -= 1;
    sr += temp; // add current carry value into sr
void VirtualMachine::setRightCarryBit()
    if (sr & 0x1) // check if sr carry bit is 1, if so, subtract 1 from it to reset to 0
    sr += 1; // add current carry value into sr
int VirtualMachine::getCarryBit()
    return (sr & 0x1);
void VirtualMachine::setOverflow()
    if (!(sr & 0x10))
        sr += 16;
void VirtualMachine::setGreaterThan()
    if (!(sr & 0x2))
        sr += 2;
int VirtualMachine::getGreaterThan()
```

```
int temp = sr \& 0x2;
   temp = temp >> 1;
    return temp;
void VirtualMachine::resetGreaterThan()
    sr = sr & 0xfffffffd;
void VirtualMachine::setLessThan()
    if (!(sr & 0x8))
        sr += 8;
int VirtualMachine::getLessThan()
    int temp = sr \& 0x8;
    temp = temp >> 3;
    return temp;
void VirtualMachine::resetLessThan()
    sr = sr & 0xffffffff7;
void VirtualMachine::setEqualTo()
    if (!(sr & 0x4))
        sr += 4;
int VirtualMachine::getEqualTo()
    int temp = sr \& 0x4;
    temp = temp >> 2;
    return temp;
void VirtualMachine::resetEqualTo()
    sr = sr & 0xfffffffb;
void VirtualMachine::load() // r[RD] = mem[ADDR]
    if (finalcode.f2.I == 0) {
        if (!isItInMemory(finalcode.f2.ADDR))
        {
              sr = sr \mid 0x400;
             pc--;
              pc = physicalToLogical(pc);
              pageNumberToGet = pageNumber(finalcode.f2.ADDR);
              execute = false;
             return;
        }
        if (!strcmp(originalFileName.c_str(), "subVector.o")) {
         }
        int temp1 = pageNumber(finalcode.f2.ADDR);
        int temp2 = offsetNumber(finalcode.f2.ADDR);
         int temp3 = frames[temp1] * FRAME_SIZE + temp2;
```

```
clock += 4;
        reg[finalcode.f2.RD] = mem[temp3];
    }
    else
        loadi();
}
void VirtualMachine::loadi() // r[RD] = CONST
{
    clock += 1;
    reg[finalcode.f3.RD] = finalcode.f3.CONST;
void VirtualMachine::store() // mem[ADDR] = r[RD]
        if (!isItInMemory(finalcode.f2.ADDR))
             sr = sr \mid 0x400;
             pc--;
             pc = physicalToLogical(pc);
             pageNumberToGet = pageNumber(finalcode.f2.ADDR);
             execute = false;
             return;
        }
        int temp1 = pageNumber(finalcode.f2.ADDR);
        int temp2 = offsetNumber(finalcode.f2.ADDR);
        int temp3 = frames[temp1] * FRAME SIZE + temp2;
        clock += 4;
        mem[temp3] = reg[finalcode.f2.RD];
        modified[temp1] = 1;
void VirtualMachine::add() // r[RD] = r[RD] + r[RS]
    clock += 1;
    if (finalcode.f1.I == 0) {
        int temp = reg[finalcode.f1.RD] + reg[finalcode.f1.RS];
        if (reg[finalcode.f1.RD] >= 0 && reg[finalcode.f1.RS] >= 0 && temp < 0)
            setOverflow();
        else if (reg[finalcode.f1.RD] < 0 && reg[finalcode.f1.RS] < 0 && temp >= 0)
            setOverflow();
        reg[finalcode.f1.RD] = temp;
        setCarryBit();
    else
        addi();
void VirtualMachine::addi() // r[RD] = r[RD] + CONST
    clock += 1;
    int temp = req[finalcode.f3.RD] + finalcode.f3.CONST;
    if (reg[finalcode.f3.RD] >= 0 && finalcode.f3.CONST >= 0 && temp < 0)
        setOverflow();
    else if (reg[finalcode.f3.RD] < 0 \&& finalcode.f3.CONST < 0 && temp >= 0)
```

```
setOverflow();
     reg[finalcode.f3.RD] = temp;
     setCarryBit();
void VirtualMachine::addc() // r[RD] = r[RD] + r[RS] + CARRY
     clock += 1;
     if (finalcode.f1.I == 0) {
         int carry = getCarryBit();
         int temp = reg[finalcode.fl.RD] + reg[finalcode.fl.RS] + carry;
         if (reg[finalcode.f1.RD] >= 0 \&\& reg[finalcode.f1.RD] >= 0 \&\& carry >= 0 \&\& temp < 0)
             setOverflow();
         else if (reg[finalcode.f1.RD] < 0 \&& reg[finalcode.f1.RD] < 0 && carry < 0 && temp >= 0)
             setOverflow();
         reg[finalcode.f1.RD] = temp;
         setCarryBit();
    else
        addci();
void VirtualMachine::addci() // r[RD] = r[RD] + CONST + CARRY
    clock += 1;
    int carry = getCarryBit();
    int temp = reg[finalcode.f3.RD] + finalcode.f3.CONST + carry;
    if (reg[finalcode.f3.RD] >= 0 \&\& finalcode.f3.CONST >= 0 \&\& carry >= 0 \&\& temp < 0)
        setOverflow();
    else if (reg[finalcode.f3.RD] < 0 && finalcode.f3.CONST < 0 && carry < 0 && temp >= 0)
        setOverflow();
    reg[finalcode.f3.RD] = temp;
    setCarryBit();
}
void VirtualMachine::sub() // r[RD] = r[RD] - r[RS]
    clock += 1;
    if (finalcode.fl.I == 0) {
        int temp = reg[finalcode.f1.RD] - reg[finalcode.f1.RS];
        int complRS = ~reg[finalcode.f1.RS] + 1;
        if (reg[finalcode.fl.RD] >= 0 \&\& complRS >= 0 \&\& temp < 0)
            setOverflow();
        else if (reg[finalcode.f1.RD] < 0 && complRS < 0 && temp >= 0)
            setOverflow();
        reg[finalcode.fl.RD] = temp;
        setCarryBit();
    }
    else
        subi();
void VirtualMachine::subi() // r[RD] = r[RD] - CONST
    clock += 1;
        int temp = reg[finalcode.f1.RD] - finalcode.f3.CONST;
```

```
int complCONST = ~finalcode.f3.CONST + 1;
        if (reg[finalcode.f1.RD] >= 0 && complCONST >= 0 && temp < 0)
            setOverflow();
        else if (reg[finalcode.f1.RD] < 0 && complCONST < 0 && temp >= 0)
            setOverflow();
        reg[finalcode.fl.RD] = temp;
        setCarryBit();
}
void VirtualMachine::subc() // r[RD] = r[RD] - r[RS] - CARRY
    clock += 1;
    if (finalcode.f1.I == 0) {
        int temp = reg[finalcode.f1.RD] - reg[finalcode.f1.RS] - getCarryBit();
        int complRS = ~reg[finalcode.f1.RS] + 1;
        if (reg[finalcode.f1.RD] >= 0 \&\& complRS >= 0 \&\& temp < 0)
            setOverflow();
        else if (reg[finalcode.f1.RD] < 0 && complRS < 0 && temp >= 0)
            setOverflow();
        reg[finalcode.f1.RD] = temp;
        setCarryBit();
    else
        subci();
void VirtualMachine::subci() // r[RD] = r[RD] - CONST - CARRY
    clock += 1;
    int temp = reg[finalcode.fl.RD] - finalcode.f3.CONST - getCarryBit();
    int complRS = ~finalcode.f3.CONST + 1;
    if (reg[finalcode.f1.RD] >= 0 \&\& complRS >= 0 \&\& temp < 0)
        setOverflow();
    else if (reg[finalcode.f1.RD] < 0 && complRS < 0 && temp >= 0)
        setOverflow();
    reg[finalcode.f1.RD] = temp;
    setCarryBit();
}
void VirtualMachine:: and() // r[RD] = r[RD] & r[RS]
    clock += 1;
    if (finalcode.f1.I == 0)
        reg[finalcode.f1.RD] = reg[finalcode.f1.RD] & reg[finalcode.f1.RS];
    else
        andi();
}
void VirtualMachine::andi() // r[RD] = r[RD] & CONST
    clock += 1;
    reg[finalcode.f1.RD] = reg[finalcode.f1.RD] & finalcode.f3.CONST;
void VirtualMachine:: xor() // r[RD] = r[RD] ^ r[RS]
{
    clock += 1;
    if (finalcode.fl.I == 0)
        reg[finalcode.f1.RD] = reg[finalcode.f1.RD] ^ reg[finalcode.f1.RS];
    else
```

```
xori();
void VirtualMachine::xori() // r[RD] = r[RD] ^ CONST
    clock += 1;
    reg[finalcode.f1.RD] = reg[finalcode.f1.RD] ^ finalcode.f3.CONST;
void VirtualMachine::_compl() // r[RD] = ~ r[RD]
    clock += 1;
    reg[finalcode.f1.RD] = ~reg[finalcode.f1.RD];
void VirtualMachine::shl() // r[RD] = r[RD] << 1, shift-in-bit = 0
    clock += 1;
    reg[finalcode.fl.RD] = reg[finalcode.fl.RD] << 1;
    setCarryBit();
void VirtualMachine::shla() // shl arithmetic Set CARRY & Sign Ext
    clock += 1;
    if (reg[finalcode.f1.RD] < 0) {</pre>
        reg[finalcode.f1.RD] = reg[finalcode.f1.RD] << 1;</pre>
        reg[finalcode.fl.RD] = reg[finalcode.fl.RD] | 0xfffff8000;
    else
       reg[finalcode.f1.RD] = reg[finalcode.f1.RD] << 1;</pre>
    setCarryBit();
void VirtualMachine::shr() // r[RD] = r[RD] >> 1, shift-in-bit = 0
    clock += 1;
    int temp = reg[finalcode.fl.RD];
    temp = temp \& 0x1;
    if (temp)
        setRightCarryBit();
    reg[finalcode.f1.RD] = reg[finalcode.f1.RD] >> 1;
}
void VirtualMachine::shra() // shr arithmetic Set CARRY & Sign Ext
    clock += 1;
    int temp = reg[finalcode.fl.RD];
    temp = temp \& 0x1;
    if (temp)
        setRightCarryBit();
    if (reg[finalcode.f1.RD] < 0) {</pre>
        reg[finalcode.f1.RD] = reg[finalcode.f1.RD] >> 1;
        reg[finalcode.f1.RD] = reg[finalcode.f1.RD] | 0xffff8000;
    }
       reg[finalcode.fl.RD] = reg[finalcode.fl.RD] >> 1;
void VirtualMachine::compr()
    clock += 1;
   if (finalcode.f1.I == 0) {
        if (reg[finalcode.fl.RD] < reg[finalcode.fl.RS]) {</pre>
```

```
setLessThan();
            resetEqualTo();
            resetGreaterThan();
        else if (reg[finalcode.fl.RD] > reg[finalcode.fl.RS]) {
            setGreaterThan();
            resetEqualTo();
            resetLessThan();
        else {
            setEqualTo();
            resetLessThan();
            resetGreaterThan();
        }
    else
        compri();
void VirtualMachine::compri()
    clock += 1;
    if (reg[finalcode.f1.RD] < finalcode.f3.CONST) {</pre>
        setLessThan();
        resetEqualTo();
        resetGreaterThan();
    }
    else if (reg[finalcode.f1.RD] > finalcode.f3.CONST) {
        setGreaterThan();
        resetEqualTo();
        resetLessThan();
    }
    else if (reg[finalcode.f1.RD] == finalcode.f3.CONST) {
        setEqualTo();
        resetLessThan();
        resetGreaterThan();
    }
}
void VirtualMachine::getstat()
                                  // r[RD] = SR
    clock += 1;
    reg[finalcode.f1.RD] = sr;
void VirtualMachine::putstat() // SR = r[RD]
    clock += 1;
    sr = reg[finalcode.f1.RD];
void VirtualMachine::jump() // pc = ADDR
    if (!isItInMemory(finalcode.f2.ADDR))
    {
        sr = sr \mid 0x400;
        pc = physicalToLogical(pc);
        pageNumberToGet = pageNumber(finalcode.f2.ADDR);
        execute = false;
        return;
    }
   int temp1 = pageNumber(finalcode.f2.ADDR);
   int temp2 = offsetNumber(finalcode.f2.ADDR);
   int temp3 = frames[temp1] * FRAME_SIZE + temp2;
   clock += 1;
   pc = temp3;
   base = frames[temp1] * FRAME SIZE;
```

```
void VirtualMachine::jumpl() // if LESS == 1, pc = ADDR, else do nothing
    if (!isItInMemory(finalcode.f2.ADDR))
    {
        sr = sr \mid 0x400;
        pc--;
        pc = physicalToLogical(pc);
        pageNumberToGet = pageNumber(finalcode.f2.ADDR);
        execute = false;
        return;
    int temp1 = pageNumber(finalcode.f2.ADDR);
    int temp2 = offsetNumber(finalcode.f2.ADDR);
    int temp3 = frames[temp1] * FRAME_SIZE + temp2;
    clock += 1;
    if(getLessThan()) {
        pc = temp3;
        base = frames[temp1] * FRAME SIZE;
    }
}
void VirtualMachine::jumpe() // if EQUAL == 1, pc = ADDR, else do nothing
    if (!isItInMemory(finalcode.f2.ADDR))
    {
        sr = sr \mid 0x400;
        pc--;
        pc = physicalToLogical(pc);
        pageNumberToGet = pageNumber(finalcode.f2.ADDR);
        execute = false;
        return;
    }
    int temp1 = pageNumber(finalcode.f2.ADDR);
    int temp2 = offsetNumber(finalcode.f2.ADDR);
    int temp3 = frames[temp1] * FRAME SIZE + temp2;
    if(getEqualTo())
        pc = temp3;
        base = frames[temp1] * FRAME SIZE;
void VirtualMachine::jumpg() // if GREATER == 1, pc = ADDR, else do nothing
    if (!isItInMemory(finalcode.f2.ADDR))
    {
        sr = sr \mid 0x400;
        pc--;
        pc = physicalToLogical(pc);
        pageNumberToGet = pageNumber(finalcode.f2.ADDR);
        execute = false;
        return;
    }
    int temp1 = pageNumber(finalcode.f2.ADDR);
    int temp2 = offsetNumber(finalcode.f2.ADDR);
    int temp3 = frames[temp1] * FRAME_SIZE + temp2;
    clock += 1;
    if(getGreaterThan())
        {
```

```
pc = temp3;
       base = frames[temp1] * FRAME SIZE;
    }
void VirtualMachine::call() // push VM status; pc = ADDR
    if (!isItInMemory(finalcode.f2.ADDR))
    {
        sr = sr \mid 0x400;
        pc--;
        pc = physicalToLogical(pc);
        pageNumberToGet = pageNumber(finalcode.f2.ADDR);
        execute = false;
        return;
    }
    int oldPc = pc;
    int temp1 = pageNumber(finalcode.f2.ADDR);
    int temp2 = offsetNumber(finalcode.f2.ADDR);
    int temp3 = frames[temp1] * FRAME SIZE + temp2;
    pc = temp3;
    base = frames[temp1] * FRAME SIZE;
    clock += 4;
    int newSp = sp - 6;
    int frameNumberToOverwrite = pageNumber(newSp);
    if (strcmp(inverted[frameNumberToOverwrite].c str(),"NULL"))
        copyFromMemory(frameNumberToOverwrite,osJobs);
    }
     inverted[frameNumberToOverwrite] = "Stack";
    mem[--sp] = sr;
    mem[--sp] = reg[0];
    mem[--sp] = reg[1];
    mem[--sp] = reg[2];
    mem[--sp] = reg[3];
    mem[--sp] = oldPc;
void VirtualMachine:: return() // pop and restore VM status
    clock += 4;
    if (sp==256) {
        returnCode(4);
        execute=false;
        return;
    inverted[pageNumber(sp)] = "NULL";
    pc = mem[sp++];
    reg[3] = mem[sp++];
    reg[2] = mem[sp++];
    reg[1] = mem[sp++];
    reg[0] = mem[sp++];
```

```
sr = mem[sp++];
    int temp1 = pageNumber(pc);
    base = temp1 * FRAME SIZE;
}
void VirtualMachine::read() // read new content of r[RD] from .in file
    returnCode(6);
    readOrWrite(finalcode.f1.RD);
    execute = false;
void VirtualMachine::write() // write r[RD] into .out file
    returnCode(7);
    readOrWrite(finalcode.f1.RD);
    execute = false;
void VirtualMachine::halt() // halt execution
    int temp1 = pageNumber(pc - 1);
    clock += 1;
    returnCode(1);
    execute = false;
}
void VirtualMachine::noop() // no operation
    clock += 1;
void VirtualMachine::returnCode(int i)
    sr &= 0xffffffff;
    int temp = i << 5;
    sr \mid = temp;
void VirtualMachine::readOrWrite(int i)
    sr &= 0xfffffcff;
    int temp = i << 8;
    sr |= temp;
 int VirtualMachine::pageNumber(int addr)
         int page = addr & 0xf8;
         return page >> 3;
int VirtualMachine::offsetNumber(int addr)
        return addr & 0x7;
    }
bool VirtualMachine::isItInMemory(int ADDR)
    int temp = pageNumber(ADDR);
```

```
if(valid[temp] == 1)
        return true;
    return false;
}
int VirtualMachine::physicalToLogical(int ADDR)
    int frame = pageNumber(ADDR);
    int offset = offsetNumber(ADDR);
    int page = 0;
    for (int i = 0; i < frames.size(); i++)
        if (frames[i] == frame) {
            page = i;
            break;
        }
    }
    page = page << 3;</pre>
    page = page + offset;
   return page;
}
```

```
* James Small & William Elder
 * Assembler.h
 * 4/11/13
 * This program simultes an assembler, translating assembly instructions into
 * binary instruction format.
 #ifndef ASSEMBLER H
#define ASSEMBLER H
#include <iostream>
#include <fstream>
#include <string>
#include <sstream>
#include <map>
#include <cstdlib>
#include <stdexcept>
using namespace std;
class NullPointerException: public runtime_error {
public:
     NullPointerException(): runtime_error("Invalid Operand Read") { }
class Assembler{
private:
   //typedefs
   typedef void(Assembler::*FP)(string);
   // Utility Functions
   void checkRD(int RD);
   void checkRS(int RS);
   void checkADDR(int ADDR);
   void checkCONST(int CONST);
   void write(int objcode);
   // Instruction Functions
   void load(string s);
   void loadi(string s);
   void store(string s);
   void add(string s);
   void addi(string s);
   void addc(string s);
   void addci(string s);
   void sub(string s);
   void subi(string s);
   void subc(string s);
   void subci(string s);
   void and(string s);
   void andi(string s);
   void xor(string s);
   void xori(string s);
   void compl(string s);
   void shl(string s);
   void shla(string s);
   void shr(string s);
   void shra(string s);
   void compr(string s);
   void compri(string s);
   void getstat(string s);
   void putstat(string s);
   void jump(string s);
   void jumpl(string s);
   void jumpe(string s);
   void jumpg(string s);
   void call(string s);
   void return(string s);
   void read(string s);
   void write(string s);
```

```
void halt(string s);
void noop(string s);

//variables
string inputFileName, outputFileName;
map<string, FP> instr;

public:
    Assembler();
    void assemble(string filename);
};

#endif
```

```
* James Small & William Elder
 * Assembler.cpp
 * 4/11/13
 * This program simultes an assembler, translating assembly instructions into
 * binary instruction format.
 #include "Assembler.h"
Assembler::Assembler()
   instr["load"] = &Assembler::load;
   instr["loadi"] = &Assembler::loadi;
   instr["store"] = &Assembler::store;
   instr["add"] = &Assembler::add;
   instr["addi"] = &Assembler::addi;
   instr["addc"] = &Assembler::addc;
   instr["addci"] = &Assembler::addci;
   instr["sub"] = &Assembler::sub;
   instr["subi"] = &Assembler::subi;
   instr["subc"] = &Assembler::subc;
   instr["subci"] = &Assembler::subci;
   instr["and"] = &Assembler:: and;
   instr["andi"] = &Assembler::andi;
   instr["xor"] = &Assembler:: xor;
   instr["xori"] = &Assembler::xori;
   instr["compl"] = &Assembler:: compl;
   instr["shl"] = &Assembler::shl;
   instr["shla"] = &Assembler::shla;
   instr["shr"] = &Assembler::shr;
   instr["shra"] = &Assembler::shra;
   instr["compr"] = &Assembler::compr;
   instr["compri"] = &Assembler::compri;
   instr["getstat"] = &Assembler::getstat;
   instr["putstat"] = &Assembler::putstat;
   instr["jump"] = &Assembler::jump;
   instr["jumpl"] = &Assembler::jumpl;
   instr["jumpe"] = &Assembler::jumpe;
   instr["jumpg"] = &Assembler::jumpg;
instr["call"] = &Assembler::call;
   instr["return"] = &Assembler:: return;
   instr["read"] = &Assembler::read;
   instr["write"] = &Assembler::write;
   instr["halt"] = &Assembler::halt;
   instr["noop"] = &Assembler::noop;
void Assembler::assemble(string filename)
   string line, opcode;
   // Set File Names for Output and Input
   inputFileName=filename;
   outputFileName = filename.erase(filename.length() -2, 2) + ".o";
   // Create Output .o File
   fstream output;
   output.open(outputFileName.c_str(), ios::out);
   output.close();
   // Open .s file for input
   fstream assemblyProq;
   assemblyProg.open(inputFileName.c_str(), ios::in);
   if (!assemblyProg.is open()) {
       cout << inputFileName << " failed to open.\n";</pre>
       exit(1);
   }
```

```
getline(assemblyProg, line);
    while (!assemblyProg.eof()) {
        if (line[0] == '!') {
            getline(assemblyProg, line);
            continue;
        istringstream str(line.c str());
        try {
            str >> opcode;
            if (not instr[opcode] )
                throw NullPointerException();
                 (this->*instr[opcode])(line.c str());
      catch (NullPointerException e) {
            cerr << e.what() << endl;</pre>
            exit(1);
        getline(assemblyProg, line);
    assemblyProg.close();
void Assembler::write(int objcode)
    fstream fout;
    fout.open(outputFileName.c str(), ios::in | ios::out | ios::ate);
    if (!fout.is open()) {
        cout << outputFileName << " failed to open.\n";</pre>
        exit(1);
    fout << objcode << endl;
    fout.close();
void Assembler::checkRD(int RD)
    if (RD > 3 | | RD < 0) {
        cout << "RD value out of range" << endl;</pre>
        exit(1);
    }
void Assembler::checkRS(int RS)
{
    if (RS > 3 | | RS < 0) {
        cout << "RS value out of range" << endl;</pre>
        exit(1);
    }
void Assembler::checkADDR(int ADDR)
    if (ADDR > 255 || ADDR < 0) {
        cout << "ADDR value out of range" << endl;</pre>
        exit(1);
    }
void Assembler::checkCONST(int CONST)
{
    if (CONST > 127 | | CONST < -128) {
        cout << "CONST value out of range" << endl;</pre>
        exit(1);
    }
void Assembler::load(string s)
```

```
int RD, ADDR;
    string OPCODE;
    istringstream stream(s.c_str());
    stream >> OPCODE >> RD >> ADDR; // Don't need OPCODE
    checkRD(RD);
    checkADDR(ADDR);
    int finalnum = 0;
    finalnum = finalnum << 11;</pre>
    finalnum += RD << 9;</pre>
    finalnum += ADDR;
   write(finalnum);
}
void Assembler::loadi(string s)
    int RD, CONST, I=1;
    string OPCODE;
    istringstream stream(s.c str());
    stream >> OPCODE >> RD >> CONST; // Don't need OPCODE
    checkRD(RD);
    checkCONST(CONST);
    int finalnum = 0;
    finalnum = finalnum << 11;</pre>
    finalnum += RD << 9;
    finalnum += I << 8;
    if (CONST < 0)
        CONST = CONST & 0x00FF;
    finalnum += CONST;
    write(finalnum);
}
void Assembler::store(string s)
    int RD, ADDR;
    string OPCODE;
    istringstream stream(s.c str());
    stream >> OPCODE >> RD >> ADDR; // Don't need OPCODE
    checkRD(RD);
    checkADDR(ADDR);
    int finalnum = 1;
    finalnum = finalnum << 11;</pre>
    finalnum += RD << 9;
    finalnum += ADDR;
    write(finalnum);
}
void Assembler::add(string s)
    int RD, RS, I = 0;
    string OPCODE;
    istringstream stream(s.c str());
    stream >> OPCODE >> RD >> RS; // Don't need OPCODE
    checkRD(RD);
    checkRS(RS);
```

```
int finalnum = 2;
    finalnum = finalnum << 11;</pre>
    finalnum += RD << 9;
    finalnum += I << 8;
    finalnum += RS << 6;
    write(finalnum);
}
void Assembler::addi(string s)
    int RD, CONST, I = 1;
    string OPCODE;
    istringstream stream(s.c_str());
    stream >> OPCODE >> RD >> CONST; // Don't need OPCODE
    checkRD(RD);
    checkCONST(CONST);
    int finalnum = 2;
    finalnum = finalnum << 11;</pre>
    finalnum += RD << 9;
    finalnum += I << 8;
    if (CONST < 0)
        CONST = CONST & 0 \times 00 FF;
    finalnum += CONST;
    write(finalnum);
void Assembler::addc(string s)
    int RD, RS, I = 0;
    string OPCODE;
    istringstream stream(s.c_str());
    stream >> OPCODE >> RD >> RS; // Don't need OPCODE
    checkRD(RD);
    checkRS(RS);
    int finalnum = 3;
    finalnum = finalnum << 11;
    finalnum += RD << 9;
    finalnum += I << 8;
    finalnum += RS << 6;
    write(finalnum);
}
void Assembler::addci(string s)
    int RD, CONST, I = 1;
    string OPCODE;
    istringstream stream(s.c str());
    stream >> OPCODE >> RD >> CONST; // Don't need OPCODE
    checkRD(RD);
    checkCONST (CONST);
    int finalnum = 3;
    finalnum = finalnum << 11;</pre>
    finalnum += RD << 9;
    finalnum += I << 8;
   if (CONST < 0)
        CONST = CONST & 0 \times 00 FF;
```

```
finalnum += CONST;
   write(finalnum);
void Assembler::sub(string s)
    int RD, RS, I = 0;
    string OPCODE;
    istringstream stream(s.c str());
    stream >> OPCODE >> RD >> RS; // Don't need OPCODE
    checkRD(RD);
    checkRS(RS);
    int finalnum = 4;
    finalnum = finalnum << 11;</pre>
    finalnum += RD << 9;
    finalnum += I << 8;
    finalnum += RS << 6;
    write(finalnum);
void Assembler::subi(string s)
    int RD, CONST, I = 1;
    string OPCODE;
    istringstream stream(s.c str());
    stream >> OPCODE >> RD >> CONST; // Don't need OPCODE
    checkRD(RD);
    checkCONST(CONST);
    int finalnum = 4;
    finalnum = finalnum << 11;</pre>
    finalnum += RD << 9;
    finalnum += I << 8;</pre>
    if (CONST < 0)
        CONST = CONST & Ox00FF;
    finalnum += CONST;
    write(finalnum);
}
void Assembler::subc(string s)
    int RD, RS, I = 0;
    string OPCODE;
    istringstream stream(s.c str());
    stream >> OPCODE >> RD >> RS; // Don't need OPCODE
    checkRD(RD);
    checkRS(RS);
    int finalnum = 5;
    finalnum = finalnum << 11;</pre>
    finalnum += RD << 9;
    finalnum += I << 8;
    finalnum += RS << 6;
    write(finalnum);
}
void Assembler::subci(string s)
    int RD, CONST, I = 1;
```

```
string OPCODE;
    istringstream stream(s.c_str());
    stream >> OPCODE >> RD >> CONST; // Don't need OPCODE
    checkRD(RD);
    checkCONST(CONST);
    int finalnum = 5;
    finalnum = finalnum << 11;</pre>
    finalnum += RD << 9;
    finalnum += I << 8;</pre>
    if (CONST < 0)
        CONST = CONST & 0 \times 00 FF;
    finalnum += CONST;
    write(finalnum);
void Assembler:: and(string s)
    int RD, RS, I = 0;
    string OPCODE;
    istringstream stream(s.c str());
    stream >> OPCODE >> RD >> RS; // Don't need OPCODE
    checkRD(RD);
    checkRS(RS);
    int finalnum = 6;
    finalnum = finalnum << 11;</pre>
    finalnum += RD << 9;
    finalnum += I << 8;
    finalnum += RS << 6;</pre>
    write(finalnum);
void Assembler::andi(string s)
    int RD, CONST, I = 1;
    string OPCODE;
    istringstream stream(s.c_str());
    stream >> OPCODE >> RD >> CONST; // Don't need OPCODE
    checkRD(RD);
    checkCONST(CONST);
    int finalnum = 6;
    finalnum = finalnum << 11;</pre>
    finalnum += RD << 9;
    finalnum += I << 8;
    if (CONST < 0)
        CONST = CONST & 0 \times 00 FF;
    finalnum += CONST;
    write(finalnum);
void Assembler:: xor(string s)
    int RD, RS, I = 0;
    string OPCODE;
    istringstream stream(s.c_str());
    stream >> OPCODE >> RD >> RS; // Don't need OPCODE
```

```
checkRD(RD);
     checkRS(RS);
     int finalnum = 7;
     finalnum = finalnum << 11;</pre>
     finalnum += RD << 9;</pre>
     finalnum += I << 8;</pre>
     finalnum += RS << 6;
     write(finalnum);
}
void Assembler::xori(string s)
     int RD, CONST, I = 1;
     string OPCODE;
     istringstream stream(s.c str());
    stream >> OPCODE >> RD >> CONST; // Don't need OPCODE
    checkRD(RD);
    checkCONST(CONST);
    int finalnum = 7;
    finalnum = finalnum << 11;</pre>
    finalnum += RD << 9;
    finalnum += I << 8;</pre>
    if (CONST < 0)
         CONST = CONST & 0 \times 00 FF;
    finalnum += CONST;
    write(finalnum);
}
void Assembler:: compl(string s)
    int RD, I = 0;
    string OPCODE;
    istringstream stream(s.c str());
    stream >> OPCODE >> RD; // Don't need OPCODE
    checkRD(RD);
    int finalnum = 8;
    finalnum = finalnum << 11;
    finalnum += RD << 9;</pre>
    write(finalnum);
}
void Assembler::shl(string s)
    int RD, I = 0;
    string OPCODE;
    istringstream stream(s.c str());
    stream >> OPCODE >> RD; // Don't need OPCODE
    checkRD(RD);
    int finalnum = 9;
    finalnum = finalnum << 11;</pre>
    finalnum += RD << 9;
    write(finalnum);
}
void Assembler::shla(string s)
```

```
{
    istringstream stream(s.c_str());
    int RD, I=0;
    string OPCODE;
    stream>>OPCODE>>RD; // Don't need OPCODE
    checkRD(RD);
    int finalnum = 10;
    finalnum = finalnum<<11;</pre>
    finalnum += RD<<9;
    write(finalnum);
}
void Assembler::shr(string s)
    int RD;
    string OPCODE;
    istringstream stream(s.c_str());
    stream >> OPCODE >> RD; // Don't need OPCODE
    checkRD(RD);
    int finalnum = 11;
    finalnum = finalnum << 11;</pre>
    finalnum += RD << 9;
    write(finalnum);
void Assembler::shra(string s)
    int RD;
    string OPCODE;
    istringstream stream(s.c str());
    stream >> OPCODE >> RD; // Don't need OPCODE
    checkRD(RD);
    int finalnum = 12;
    finalnum = finalnum << 11;</pre>
    finalnum += RD << 9;
    write(finalnum);
}
void Assembler::compr(string s)
    int RD, RS;
    string OPCODE;
    istringstream stream(s.c str());
    stream >> OPCODE >> RD >> RS; // Don't need OPCODE
    checkRD(RD);
    checkRS(RS);
    int finalnum = 13;
    finalnum = finalnum << 11;</pre>
    finalnum += RD << 9;
    finalnum += RS << 6;
    write(finalnum);
void Assembler::compri(string s)
    int RD, I = 1, CONST;
    string OPCODE;
    istringstream stream(s.c_str());
    stream >> OPCODE >> RD >> CONST; // Don't need OPCODE
```

```
checkRD(RD);
     checkCONST(CONST);
     int finalnum = 13;
     finalnum = finalnum << 11;</pre>
     finalnum += RD << 9;
     finalnum += I << 8;
     if (CONST < 0)
         CONST = CONST & 0 \times 00 FF;
     finalnum += CONST;
    write(finalnum);
void Assembler::getstat(string s)
    int RD;
    string OPCODE;
    istringstream stream(s.c_str());
stream >> OPCODE >> RD; // Don't need OPCODE
    checkRD(RD);
    int finalnum = 14;
    finalnum = finalnum << 11;
    finalnum += RD << 9;</pre>
    write(finalnum);
void Assembler::putstat(string s)
    int RD;
    string OPCODE;
    istringstream stream(s.c_str());
    stream >> OPCODE >> RD; // Don't need OPCODE
    checkRD(RD);
    int finalnum = 15;
    finalnum = finalnum << 11;</pre>
    finalnum += RD << 9;
    write(finalnum);
}
void Assembler::jump(string s)
    int ADDR;
    string OPCODE;
    istringstream stream(s.c_str());
    stream >> OPCODE >> ADDR; // Don't need OPCODE
    checkADDR (ADDR);
    int finalnum=16;
    finalnum = finalnum << 11;</pre>
    finalnum += ADDR;
    write(finalnum);
void Assembler::jumpl(string s)
    int ADDR;
    string OPCODE;
```

```
istringstream stream(s.c str());
    stream >> OPCODE >> ADDR; // Don't need OPCODE
    checkADDR (ADDR);
    int finalnum = 17;
    finalnum = finalnum << 11;</pre>
    finalnum += ADDR;
    write(finalnum);
void Assembler::jumpe(string s)
    int ADDR;
    string OPCODE;
    istringstream stream(s.c str());
    stream >> OPCODE >> ADDR; // Don't need OPCODE
    checkADDR (ADDR);
    int finalnum = 18;
    finalnum = finalnum << 11;
    finalnum += ADDR;
    write(finalnum);
void Assembler::jumpg(string s)
    int ADDR;
    string OPCODE;
    istringstream stream(s.c str());
    stream >> OPCODE >> ADDR; // Don't need OPCODE
    checkADDR (ADDR);
    int finalnum = 19;
    finalnum = finalnum << 11;</pre>
    finalnum += ADDR;
    write(finalnum);
}
void Assembler::call(string s)
    int ADDR;
    string OPCODE;
    istringstream stream(s.c_str());
    stream >> OPCODE >> ADDR; // Don't need OPCODE
    checkADDR (ADDR);
    int finalnum = 20;
    finalnum = finalnum << 11;
    finalnum += ADDR;
    write(finalnum);
void Assembler:: return(string s)
    int finalnum = 21;
    write(finalnum << 11);</pre>
void Assembler::read(string s)
```

```
int RD;
    string OPCODE;
    istringstream stream(s.c str());
    stream >> OPCODE >> RD; // Don't need OPCODE
    checkRD(RD);
    int finalnum = 22;
    finalnum = finalnum << 11;</pre>
    finalnum += RD << 9;</pre>
    write(finalnum);
}
void Assembler::write(string s)
    int RD;
    string OPCODE;
    istringstream stream(s.c str());
    stream >> OPCODE >> RD; // Don't need OPCODE
    checkRD(RD);
    int finalnum = 23;
    finalnum = finalnum << 11;</pre>
    finalnum += RD << 9;
    write(finalnum);
void Assembler::halt(string s)
    int finalnum = 24;
    write(finalnum << 11);</pre>
}
void Assembler::noop(string s)
    int finalnum = 25;
    write(finalnum << 11);</pre>
}
```

```
* James Small & William Elder
  PCB.h
 * 5/7/13
 * This is the process control block class to be used in our OS. We keep a list
 * of pointers to these PCB's in order to hold information about all current
 ^{\star} running processes on the system. The information about the processes is copied
 ^{\star} to and from these PCB's in order to facilitate our os simulation.
 #ifndef PCB H
#define PCB H
using namespace std;
class PCB {
private:
   int pc,sr,sp,base,limit;
    static const int REG FILE SIZE = 4;
    static const int FRAME COUNT = 32;
   vector < int > reg;
   string readfilename, writefilename, originalfilename, stfilename, state;
   fstream readIntoRegister;
   fstream writeToRegister;
    int cpuTime, waitingTime, turnaroundTime, contextSwitchTime, idleTime,
ioTime, largestStackSize, interruptTime;
    vector <int> frames;
    vector <int> valid;
   vector <int> modified;
public:
   PCB() {
       reg = vector <int> (REG FILE SIZE);
       pc = sr = cpuTime = waitingTime = turnaroundTime = ioTime = largestStackSize =
contextSwitchTime = idleTime = 0;
       for (int i = 0;i < REG FILE SIZE; i++) {</pre>
           reg[i] = 0;
       sp = 256;
       state = "new";
       frames = vector <int> (FRAME COUNT);
       valid = vector <int> (FRAME_COUNT);
       modified = vector <int> (FRAME COUNT);
       for (int i = 0; i < FRAME COUNT; i++)
           frames[i] = valid[i] = modified[i] = 0;
   };
   void openReadInFileStream() {
       readIntoRegister.open(readfilename.c_str(), ios::in);
       if (!readIntoRegister.is_open()) {
          // exit(1);
       }
   }
   void openWriteOutFileStream() {
       writeToRegister.open(writefilename.c_str(), ios::out);
       if (!writeToRegister.is open()) {
           cout << writefilename << " failed to open.\n";</pre>
           exit(1);
       }
   }
```

```
~PCB() {
    readIntoRegister.close();
    writeToRegister.close();
}

friend class OS;
  friend class VirtualMachine;
};
#endif
```

```
! read in first vector
read 0
store 0 61
read 0
store 0 62
read 0
store 0 63
read 0
store 0 64
read 0
store 0 65
read 0
store 0 66
read 0
store 0 67
read 0
store 0 68
read 0
store 0 69
read 0
store 0 70
! read in second vector and add
load 0 61
     1
read
add
      0 1
write 0
load 0 62
     1
read
      0 1
add
write 0
load 0 63
     1
read
add
      0 1
write 0
load 0 64
read
     1
add
      0 1
write 0
load 0 65
read 1
add
      0 1
write 0
load 0 66
read 1
add
      0 1
write 0
load 0 67
read 1
add
      0 1
write 0
load 0 68
read 1
add
      0 1
write 0
load 0 69
read 1
add
      0 1
write 0
load 0 70
read
     1
add
      0 1
write 0
halt
noop
noop
noop
noop
noop
noop
noop
noop
```

noop

add Vector. 5

add Vectoriin

```
45056
2109
45056
2110
45056
2111
45056
2112
45056
2113
45056
2114
45056
2115
45056
2116
45056
2117
45056
2118
61
45568
4160
47104
62
45568
4160
47104
63
45568
4160
47104
64
45568
4160
47104
65
45568
4160
47104
66
45568
4160
47104
67
45568
4160
47104
68
45568
4160
47104
69
45568
4160
47104
70
45568
4160
47104
49152
10
10
10
10
10
10
```

```
3
6
12
15
18
21
13
20
23
6
20
24
28
20
20
20
20
20
20
20
20
20
20
```

CPU Time = 299 cycles Waiting Time = 1913 cycles Turnaround Time = 5144 cycles IO Time = 1998 cycles Largest Stack Size = 0

System Specific Accounting Data

System Time = 4066 cycles System CPU Utilization = 80.3254% User CPU Utilization = 49.8767% ThroughPut = 12.3274 processes per second addrector.out

add vector ont fito

Process Specific Accounting Data

CPU Time = 432 cycles Waiting Time = 3474 cycles Turnaround Time = 10450 cycles IO Time = 2862 cycles Largest Stack Size = 0

System Specific Accounting Data

System Time = 8117 cycles System CPU Utilization = 74.2675% User CPU Utilization = 46.6759% ThroughPut = 6.56944 processes per second

```
! main for factorial program
                                                                     fact1.5
       loadi 0.1 ! line 0, R0 = fact(R1)
                     ! input R1
       read
            1
       call
                     ! call fact
             6
       load 0 33
                     ! receive result of fact
       write 0
       halt
! fact function
       compri 1 1
                     ! line 6
       jumpe 14
                     ! jump over the recursive call to fact if
       jumpl 14
                     ! R1 is less than or equal 1
                     ! call mult (R0 = R0 * R1)
       call
             16
       load
             0 34
                     ! receive result of mult
                     ! decrement multiplier (R1) and multiply again
       subi
            1 1
       call
            6
                     ! call fact
       load 0 33
       store 0 33
                     ! line 14, return RO (result of fact)
       return
! mult function
       loadi 28
                     ! line 16, init R2 (counter)
       loadi 3 0
                     ! init R3 (result of mult)
                     ! line 18 (loop), shift right multiplier set CARRY
             1
       store 2 35
                     ! save counter
       getstat 2
                     ! to find CARRY's value
            2 1
       andi
       compri 2 1
       jumpe 25
                     ! if CARRY==1 add
       jump
             26
                     ! otherwise do nothing
       add
             3 0
       shl
             0
                     ! make multiplicand ready for next add
       load 2 35
                     ! restore counter
             2 1
                     ! decrement counter
       compri 2 0
                     ! if counter > 0 jump to loop
       jumpa 18
                     ! return R3 (result of mult)
       store 3 34
       return
                     ! line 33, fact return value
       noop
                     ! line 34, mult return value
       noop
       noop
                     ! line 35, mult counter
```

 factl.0

CPU Time = 1057 cycles Waiting Time = 3821 cycles Turnaround Time = 7571 cycles IO Time = 54 cycles Largest Stack Size = 36

System Specific Accounting Data

System Time = 4066 cycles
System CPU Utilization = 80.3254%
User CPU Utilization = 49.8767%
ThroughPut = 12.3274 processes per second

factl.ont

CPU Time = 2192 cycles Waiting Time = 6494 cycles Turnaround Time = 13987 cycles IO Time = 162 cycles Largest Stack Size = 84

System Specific Accounting Data

System Time = 8117 cycles
System CPU Utilization = 74.2675% User CPU Utilization = 46.6759% ThroughPut = 6.56944 processes per second factlout Fifo

```
! main for factorial program
       loadi 0 1
                    ! line 0, R0 = fact(R1)
                                                                      fact2.5
                      ! input R1
              1
       call
              6
                      ! call fact
       load
              0 33
                      ! receive result of fact
       write 0
       halt
! fact function
       compri 1 1
                      ! line 6
                      ! jump over the recursive call to fact if
       jumpe 14
                      ! R1 is less than or equal 1
       jumpl
              14
       call
              16
                      ! call mult (R0 = R0 * R1)
              0 34
                      ! receive result of mult
       load
       subi
              1 1
                      ! decrement multiplier (R1) and multiply again
              6
                      ! call fact
       call
              0 33
       load
       store 0 33
                      ! line 14, return RO (result of fact)
       return
! mult function
       loadi 28
                      ! line 16, init R2 (counter)
       loadi 3 0
                      ! init R3 (result of mult)
       shr
              1
                      ! line 18 (loop), shift right multiplier set CARRY
       store 2 35
                      ! save counter
                      ! to find CARRY's value
       getstat 2
              2 1
       andi
       compri 2 1
       jumpe 25
                      ! if CARRY==1 add
                      ! otherwise do nothing
              26
        jump
              3 0
       add
                      ! make multiplicand ready for next add
       shl
              0
       load
              2 35
                      ! restore counter
                      ! decrement counter
       subi
              2 1
       compri 2 0
                      ! if counter > 0 jump to loop
       jumpg 18
       store 3 34
                      ! return R3 (result of mult)
       return
                      ! line 33, fact return value
       noop
                      ! line 34, mult return value
       noop
                       ! line 35, mult counter
       noop
```

factziin

factz. 0

CPU Time = 1471 cycles Waiting Time = 3809 cycles Turnaround Time = 7593 cycles IO Time = 54 cycles Largest Stack Size = 48

System Specific Accounting Data

System Time = 4066 cycles System CPU Utilization = 80.3254% User CPU Utilization = 49.8767% ThroughPut = 12.3274 processes per second factz.ont

CPU Time = 2800 cycles
Waiting Time = 6478 cycles
Turnaround Time = 14039 cycles
IO Time = 108 cycles
Largest Stack Size = 90

System Specific Accounting Data

System Time = 8117 cycles
System CPU Utilization = 74.2675%
User CPU Utilization = 46.6759%
ThroughPut = 6.56944 processes per second

fact 2.out

| loadi  | 0 | 0 | ţ | i = 0           |
|--------|---|---|---|-----------------|
| compri | 0 | 6 | ţ | 6 pairs to read |
| jumpe  | 9 |   | ! | i == 6 done     |
| 1000   | 1 |   |   |                 |
| read   | 2 |   |   |                 |
|        | 1 | 2 |   |                 |
| write  | 1 |   |   |                 |
| addi   | 0 | 1 | ! | i++             |
| jump   | 1 |   | ŀ | loop again      |
| halt   |   |   |   |                 |

io.5

 $N_i, 0_i'$ 

.

.

70.01 V11

Process Specific Accounting Data

CPU Time = 58 cycles Waiting Time = 3289 cycles Turnaround Time = 7292 cycles IO Time = 486 cycles Largest Stack Size = 0

System Specific Accounting Data

System Time = 4066 cycles System CPU Utilization = 80.3254% User CPU Utilization = 49.8767% ThroughPut = 12.3274 processes per second

10.ont

Process Specific Accounting Data

CPU Time = 58 cycles Waiting Time = 6202 cycles Turnaround Time = 14256 cycles IO Time = 486 cycles Largest Stack Size = 0

System Specific Accounting Data

System Time = 8117 cycles System CPU Utilization = 74.2675% User CPU Utilization = 46.6759% ThroughPut = 6.56944 processes per second halt

Simple1.5

Simple1.0

CPU Time = 25 cycles
Waiting Time = 3853 cycles
Turnaround Time = 7889 cycles
IO Time = 27 cycles
Largest Stack Size = 0

System Specific Accounting Data

System Time = 4066 cycles System CPU Utilization = 80.3254% User CPU Utilization = 49.8767% ThroughPut = 12.3274 processes per second simple 1. onl

CPU Time = 25 cycles Waiting Time = 6766 cycles Turnaround Time = 14853 cycles IO Time = 27 cycles Largest Stack Size = 0

System Specific Accounting Data

System Time = 8117 cycles System CPU Utilization = 74.2675% User CPU Utilization = 46.6759% ThroughPut = 6.56944 processes per second Simplel . out f:fu

Simple2.5

Simple 2.0

CPU Time = 19 cycles
Waiting Time = 3782 cycles
Turnaround Time = 7737 cycles
IO Time = 27 cycles
Largest Stack Size = 0

System Specific Accounting Data

System Time = 4066 cycles System CPU Utilization = 80.3254% User CPU Utilization = 49.8767% ThroughPut = 12.3274 processes per second Simple2.out

CPU Time = 19 cycles Waiting Time = 6695 cycles Turnaround Time = 14701 cycles IO Time = 27 cycles Largest Stack Size = 0

System Specific Accounting Data

System Time = 8117 cycles
System CPU Utilization = 74.2675%
User CPU Utilization = 46.6759%
ThroughPut = 6.56944 processes per second

Simple2.out fifo read 0
loadi 1 -2
add 0 1 ! subtract 2 from value read
write 0
halt

5 nb. 5

5ub.in

CPU Time = 5 cycles
Waiting Time = 3634 cycles
Turnaround Time = 7517 cycles
IO Time = 54 cycles
Largest Stack Size = 0

System Specific Accounting Data

System Time = 4066 cycles System CPU Utilization = 80.3254% User CPU Utilization = 49.8767% ThroughPut = 12.3274 processes per second 5nb.ont

CPU Time = 5 cycles Waiting Time = 6547 cycles Turnaround Time = 14481 cycles IO Time = 54 cycles Largest Stack Size = 0

System Specific Accounting Data

System Time = 8117 cycles
System CPU Utilization = 74.2675%
User CPU Utilization = 46.6759%
ThroughPut = 6.56944 processes per second

Sub.out fifo

```
! read in 20 integers
read 0
store 0 121
read 0
store 0 122
read 0
store 0 123
read 0
store 0 124
read 0
store 0 125
read 0
store 0 126
read 0
store 0 127
read 0
store 0 128
read 0
store 0 129
read 0
store 0 130
read 0
store 0 131
read 0
store 0 132
read 0
store 0 133
read 0
store 0 134
read 0
store 0 135
read 0
store 0 136
read 0
store 0 137
read 0
store 0 138
read 0
store 0 139
read 0
store 0 140
! read in second vector and subtract
load 0 121
read
     1
sub
     0 1
write 0
load 0 122
read
     1
     0 1
sub
write 0
load 0 123
read
     1
     0 1
sub
write 0
load 0 124
read 1
     0 1
sub
write 0
load 0 125
read 1
     0 1
sub
write 0
load 0 126
read
     1
     0 1
sub
write 0
load 0 127
read
     1
     0 1
sub
write 0
     0 128
load
read
     1
```

subvector. 5

```
sub
       0 1
write 0
 load 0 129
read
      1
 sub
       0 1
write 0
load 0 130
read
      1
sub
       0 1
write 0
load 0 131
read 1
sub
      0 1
write 0
load 0 132
read
      1
sub
      0 1
write 0
load 0 133
read
      1
sub
      0 1
write 0
load 0 134
read
      1
sub
      0 1
write 0
load 0 135
read
      1
sub
      0 1
write 0
load 0 136
read
      1
      0 1
sub
write 0
load 0 137
read
      1
      0 1
sub
write 0
load 0 138
read
      1
      0 1
sub
write 0
load 0 139
read
      1
      0 1
sub
write 0
load 0 140
read
      1
sub
      0 1
write 0
halt
noop
```

1 2 3 4 5 6 7 8 9 10 2 4 6 8 10 12 14 16 18 20 2 4 6 8 10 12 14 16 18 20 1 2 3 4 5 6 7 8 9 10

SubVector.in

Subvector.0

subvector. O

```
-1
-2
-4
-2
0
2
4
6
8
10
-9
-8
-7
-6
-5
-4
-3
-2
-19978
8246
-18442
```

128

Process Specific Accounting Data

CPU Time = 355 cycles
Waiting Time = 1830 cycles
Turnaround Time = 4453 cycles
IO Time = 2268 cycles
Largest Stack Size = 0

System Specific Accounting Data

System Time = 4066 cycles System CPU Utilization = 80.3254% User CPU Utilization = 49.8767% ThroughPut = 12.3274 processes per second Subvertor. ont

Subventor.out fifo

Process Specific Accounting Data

CPU Time = 817 cycles
Waiting Time = 3293 cycles
Turnaround Time = 9294 cycles
IO Time = 5184 cycles
Largest Stack Size = 0

System Specific Accounting Data

System Time = 8117 cycles System CPU Utilization = 74.2675% User CPU Utilization = 46.6759% ThroughPut = 6.56944 processes per second

```
loadi 0 1  ! i = 1
  loadi 1 0  ! sum = 0
  read 2
  compr 0 2
  jumpe 8   ! done
  add 1 0  ! sum += i
  addi 0 1  ! i++
  jump 3   ! loop again
  write 1
  halt
```

sum1.5

Sumlin

Sum1.0

CPU Time = 251 cycles Waiting Time = 2557 cycles Turnaround Time = 5671 cycles IO Time = 54 cycles Largest Stack Size = 0

System Specific Accounting Data

System Time = 4066 cycles System CPU Utilization = 80.3254% User CPU Utilization = 49.8767% ThroughPut = 12.3274 processes per second Sum1.out

CPU Time = 251 cycles
Waiting Time = 5600 cycles
Turnaround Time = 12920 cycles
IO Time = 54 cycles
Largest Stack Size = 0

System Specific Accounting Data

System Time = 8117 cycles System CPU Utilization = 74.2675% User CPU Utilization = 46.6759% ThroughPut = 6.56944 processes per second 5uml.ont

```
loadi 0 1  ! i = 1
  loadi 1 0  ! sum = 0
  read 2
  compr 0 2
  jumpe 8   ! done
  add 1 0  ! sum += i
  addi 0 1  ! i++
  jump 3   ! loop again
  write 1
  halt
```

Sum2.5

Sum2.in

5nm2.0

CPU Time = 506 cycles Waiting Time = 1307 cycles Turnaround Time = 3963 cycles IO Time = 54 cycles Largest Stack Size = 0

System Specific Accounting Data

System Time = 4066 cycles
System CPU Utilization = 80.3254%
User CPU Utilization = 49.8767%
ThroughPut = 12.3274 processes per second

Sumz.out

CPU Time = 506 cycles Waiting Time = 4732 cycles Turnaround Time = 11560 cycles IO Time = 54 cycles Largest Stack Size = 0

System Specific Accounting Data

System Time = 8117 cycles System CPU Utilization = 74.2675% User CPU Utilization = 46.6759% ThroughPut = 6.56944 processes per second Sum2.our